<template>
  <div>
    <!-- 操作按钮 -->
    <!-- <el-button @click="handleAdd">代码编辑，高亮补全</el-button>
    <el-button @click="handleMerge">代码版本，差异对比</el-button> -->

    <el-form ref="form" :model="form" label-width="80px">
      <div class="in-coder-panel">
        <textarea ref="textarea" />
        <el-select v-if="ifshow" v-model="mode" class="code-mode-select" @change="changeMode">
          <!-- eslint-disable-next-line vue/no-template-shadow -->
          <el-option v-for="mode in modes" :key="mode.value" :label="mode.label" :value="mode.value" />
        </el-select>
      </div>
    </el-form>
  </div>
</template>

<script>
// 引入全局实例
import CodeMirror from 'codemirror'
// 核心样式
import 'codemirror/lib/codemirror.css'
// 引入主题后还需要在 options 中指定主题才会生效
import 'codemirror/theme/idea.css'

// 需要引入具体的语法高亮库才会有对应的语法高亮效果
// codemirror 官方其实支持通过 /addon/mode/loadmode.js 和 /mode/meta.js 来实现动态加载对应语法高亮库
// 但 vue 貌似没有无法在实例初始化后再动态加载对应 JS ，所以此处才把对应的 JS 提前引入
import 'codemirror/mode/clike/clike.js'
import 'codemirror/mode/sql/sql.js'

// 代码补全提示
import 'codemirror/addon/hint/anyword-hint.js'
import 'codemirror/addon/hint/show-hint.css'
import 'codemirror/addon/hint/show-hint.js'
import 'codemirror/addon/hint/sql-hint.js'

// 代码版本差异比较
import 'codemirror/addon/merge/merge.js'
import 'codemirror/addon/merge/merge.css'
import DiffMatchPatch from 'diff-match-patch'

window.diff_match_patch = DiffMatchPatch
window.DIFF_DELETE = -1
window.DIFF_INSERT = 1
window.DIFF_EQUAL = 0

export default {
  name: 'Code',

  props: {
    // 外部传入的内容，用于实现双向绑定
    // eslint-disable-next-line vue/require-default-prop
    value: String,
    // 外部传入的语法类型
    language: {
      type: String,
      default: null
    },
    // 展示选择
    ifshow: {
      type: Boolean,
      default: true
    },
    // eslint-disable-next-line vue/require-default-prop
    name: String
  },

  data() {
    return {
      // 内部真实的内容
      code: '',
      // 默认的语法类型
      mode: 'java',
      // 编辑器实例
      coder: null,
      // 默认配置
      options: {
        mode: 'text/x-java',
        // 缩进格式
        tabSize: 4,
        // 主题，对应主题库 JS 需要提前引入
        theme: 'idea',
        // 显示行号
        lineNumbers: true,
        line: true,
        extraKeys: { 'Ctrl': 'autocomplete' }
      },
      // 支持切换的语法高亮类型，对应 JS 已经提前引入
      // 使用的是 MIME-TYPE ，不过作为前缀的 text/ 在后面指定时写死了
      modes: [{
        value: 'x-java',
        label: 'Java'
      }, {
        value: 'x-sql',
        label: 'SQL'
      }],
      // 弹出层标题
      title: '',
      titleBBB: '',
      // 是否显示弹出层
      open: true,
      openBBB: false,
      form: {}
    }
  },
  mounted() {
    this._initialize()
  },
  methods: {
    // 初始化
    _initialize() {
      // 尝试从父容器获取语法类型
      if (this.language) {
        // 获取具体的语法类型对象

        const modeObj = this._getLanguage(this.language)
        console.log('aa', this.language, modeObj)
        // 判断父容器传入的语法是否被支持
        if (modeObj) {
          this.mode = modeObj.label
          this.options.mode = `text/${modeObj.value}`
        }
      }
      // 初始化编辑器实例，传入需要被实例化的文本域对象和默认配置
      this.coder = CodeMirror.fromTextArea(this.$refs.textarea, this.options)
      // 编辑器赋值

      const val = 'package com.bawei.etl\n' +
            '\n' +
            ` public class ${this.name}()  {\n` +
            '\n' +
            '   public void execute(Map<String, Object> params) {\n' +
            '\n' +
            '   }\n' +
            '}'
      if (this.language === 'java') {
        this.coder.setValue(this.value || val)
      } else {
        this.coder.setValue(this.value)
      }

      // this.coder.setValue(this.value || val)

      // 支持双向绑定
      this.coder.on('change', (coder) => {
        this.code = coder.getValue()

        if (this.$emit) {
          this.$emit('input', this.code)
        }
      })
    },
    // 获取当前语法类型
    _getLanguage(language) {
      // 在支持的语法类型列表中寻找传入的语法类型
      return this.modes.find((mode) => {
        // 所有的值都忽略大小写，方便比较
        const currentLanguage = language.toLowerCase()
        const currentLabel = mode.label.toLowerCase()
        const currentValue = mode.value.toLowerCase()

        // 由于真实值可能不规范，例如 java 的真实值是 x-java ，所以讲 value 和 label 同时和传入语法进行比较
        return currentLabel === currentLanguage || currentValue === currentLanguage
      })
    },
    // 更改模式
    changeMode(val) {
      // 修改编辑器的语法配置
      this.coder.setOption('mode', `text/${val}`)

      // 获取修改后的语法
      const label = this._getLanguage(val).label.toLowerCase()

      // 允许父容器通过以下函数监听当前的语法值
      this.$emit('language-change', label)
    },

    initUI(value, orig) {
      if (value == null) return
      const target = document.getElementById('view')
      target.innerHTML = ''
      CodeMirror.MergeView(target, {
        value: value, // 上次内容
        origLeft: null,
        orig: orig, // 本次内容
        lineNumbers: true, // 显示行号
        mode: 'shell',
        highlightDifferences: true,
        styleActiveLine: true,
        matchBrackets: true,
        connect: 'align',
        readOnly: true // 只读 不可修改
      })
    },

    /** 按钮操作 */
    handleAdd() {
      this.$nextTick(function() {
        this._initialize()
      })
    },

    /** 按钮操作 */
    handleUpdateBBB() {
      this.openBBB = true
      this.titleBBB = '代码版本，差异对比'

      // 初始化版本差异
      this.$nextTick(function() {
        this.initUI('package com.bawei.etl\n' +
            'public class 类名(生成)  {"\n' +
            '\n' +
            'public void execute(Map<String, Object> params) {\n' +
            '\n' +
            '}n' +
            '}')
      })
    }

  }
}
</script>
<style scoped>
  .in-coder-panel{
    flex-grow:1;
    display:flex;
    position:relative;
    border:1px solid #ccc;
  }

  .CodeMirror{
    width: 100%;
    flex-grow:1;
    z-index:1;
  }

  .CodeMirror-code{
    line-height:19px;
  }

  .code-mode-select{
    position: absolute;
    z-index: 2;
    right: 10px;
    top: 10px;
    max-width: 130px;
  }

</style>

